import random
from datetime import datetime

from info import constants,redis_store, db
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.captcha.captcha import captcha
from info.utils.response_code import RET
from . import  passport_blue
from  flask import request,current_app,make_response,json,jsonify,session
import re
# 功能描述：退出登录
# 请求路径: /passport/logout
# 请求方式: POST
# 请求参数: 无
# 返回值: errno, errmsg
@passport_blue.route('/logout',methods=["POST"])
def logout():
    #1.清除session
    session.pop("user_id",None)
    session.pop("nick_name",None)
    session.pop("mobile",None)
    session.pop("is_admin",None)
    #2.返回响应
    return jsonify(errno=RET.OK,errmsg="退出成功")
# 功能描述：登录用户
# 请求路径: /passport/login
# 请求方式: POST
# 请求参数: mobile,password
# 返回值: errno, errmsg
@passport_blue.route('/login', methods=['POST'])
def login():
    """
    1.获取请求参数
    2.校验参数，为空校验
    3.根据用户手机号取出用户对象
    4.判断用户对象是否存在
    5.校验密码是否正确
    6.记录用户登录信息到session
    7.返回响应
    :return:
    """
    # 1.获取请求参数
    mobile=request.json.get("mobile")
    password=request.json.get("password")
    # 2.校验参数，为空校验
    if not all([mobile,password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")
    # 3.根据用户手机号取出用户对象
    try:
        user=User.query.filter(User.mobile==mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")
    # 4.判断用户对象是否存在
    if not user:
        return jsonify(errno=RET.NODATA,errmsg="该用户不存在")
    # 5.校验密码是否正确
    # if user.password_hash!=password:
    if not user.check_password(password):
        return jsonify(errno=RET.DATAERR,errmsg="密码错误")
    # 6.记录用户登录信息到session
    session["user_id"]=user.id
    session["nick_name"]=user.nick_name
    session["mobile"]=user.mobile

    # 6.1记录用户最后一次登录时间
    user.last_login=datetime.now()

    # try:
    #     db.session.commit()
    # except Exception as e:
    #     current_app.logger.error(e)

    # 7.返回响应
    return jsonify(errno=RET.OK,errmsg="登录成功")
# 功能描述：注册用户
"""
请求路径: /passport/register
请求方式: POST
请求参数: mobile, sms_code,password
返回值: errno, errmsg
"""
@passport_blue.route('/register', methods=['POST'])
def register():
    """"
    1.获取请求参数(手机号, 短信验证码, 密码)
    2.校验参数, 为空校验
    3.通过手机号作为key取出redis中的短信验证码
    4.判断短信验证码是否过期
    5.删除redis中的短信验证码
    6.判断redis中的短信验证码和传入进来的短信验证码是否相等
    7.创建用户对象, 设置属性
    8.保存用户到数据库
    9.返回响应
    :return:
    """""
    # 1.获取请求参数(手机号, 短信验证码, 密码)
    # json_data=request.data
    # dict_data=json.loads(json_data)
    dict_data=request.get_json()
    mobile=dict_data.get("mobile")
    sms_code=dict_data.get("sms_code")
    password=dict_data.get("password")
    # 2.校验参数, 为空校验
    if not all([mobile,sms_code,password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")
    # 3.通过手机号作为key取出redis中的短信验证码
    try:
        redis_sms_code=redis_store.get("sms_code:%s"%mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="短信验证码获取失败")
    # 4.判断短信验证码是否过期
    if not redis_sms_code:
        return jsonify(errno=RET.NODATA,errmsg="短信验证码已过期")
    # 5.删除redis中的短信验证码
    try:
        redis_store.delete("sms_code:%s"%mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.NODATA, errmsg="短信验证码删除异常")
    # 6.判断redis中的短信验证码和传入进来的短信验证码是否相等
    if redis_sms_code!=sms_code:
        return jsonify(errno=RET.DBERR,errmsg="短信验证码填写错误")
    # 7.创建用户对象, 设置属性
    user=User()
    user.nick_name=mobile
    # user.password_hash=password
    user.password=password
    user.mobile=mobile
    user.signature="这个人很懒什么都没有"
    # 8.保存用户到数据库
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR,errmsg="注册用户失败")

    # 9.返回响应
    return jsonify(errno=RET.OK,errmsg="注册成功")

"""
发送短信验证码
请求路径: /passport/sms_code
请求方式: POST
请求参数: mobile, image_code,image_code_id
返回值: errno, errmsg
"""
@passport_blue.route('/sms_code', methods=['POST'])
def sms_code ():
    """
    1.获取请求参数
    2.校验参数，为空校验
    3.校验手机号的格式
    4.通过图片验证码编号，取出redis中的图片验证码A
    5.判断图片验证码A是否过期
    6.删除图片验证码A
    7.判断请求参数中的验证码B，和redis中的验证码A是否相等
    8.生成短信验证码（随机数)
    9.调用ccp模块发送短信
    10.判断短信是否发送成功
    11.保存短信验证码到redis
    12.返回相应
    :return:
    """""
    # 1.获取请求参数
    json_data=request.data
    dict_data=json.loads(json_data)
    mobile=dict_data.get("mobile")
    image_code=dict_data.get("image_code")
    image_code_id=dict_data.get("image_code_id")

    # 2.校验参数，为空校验
    if not all([mobile, image_code,image_code_id]):
        return jsonify(errno=RET.PARAMERR,ermsg="参数不完整")
    # 3.校验手机号的格式
    if not re.match("1[356789]\\d{9}",mobile):
        return jsonify(errno=RET.DATAERR,errmsg="手机号格式不正确")
    # 4.通过图片验证码编号，取出redis中的图片验证码A
    try:
        redis_image_code=redis_store.get("image_code:%s"%image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="图片验证码失败")
    # 5.判断图片验证码A是否过期
    if not redis_image_code:
        return jsonify(errno=RET.NODATA,errmsg="图片验证码已过期")
    # 6.删除图片验证码A
    try:
        redis_store.delete("image_code:%s"%image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,eremsg="删除图片验证码失败")
    # 7.判断请求参数中的验证码B，和redis中的验证码A是否相等
    if image_code.lower() !=redis_image_code.lower():
        return jsonify(errno=RET.DATAERR,errmsg="图片验证码填写错误")
    # 8.生成短信验证码（随机数)
    sms_code="%06d"%random.randint(0,999999)
    current_app.logger.debug("短信验证码是 = %s" % sms_code)
    # 9.调用ccp模块发送短信
    # try:
    #     ccp=CCP()
    #     result=ccp.send_template_sms(mobile,[sms_code,constants.SMS_CODE_REDIS_EXPIRES/60],1)
    # except Exception as e:
    #     current_app.logger.error(e)
    #     return jsonify(errno=RET.THIRDERR,errmsg="云通讯异常")
    #
    # # 10.判断短信是否发送成功
    # if result==-1:
    #     return jsonify(errno=RET.THIRDERR,errmsg="短信发送失败")
    # 11.保存短信验证码到redis
    try:
        redis_store.set("sms_code:%s"%mobile,sms_code,constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="短信验证码保存失败")
    # 12.返回相应
    return jsonify(errno=RET.OK,errmsg="发送成功")
# 功能描述：获取图片验证码
# 请求路径: /passport/image_code
# 请求方式: GET
# 请求参数: cur_id, pre_id
# 返回值: 图片验证码
@passport_blue.route('/image_code')
def image_code():
    # 1.获取参数，校验参数
    cur_id=request.args.get("cur_id")
    pre_id=request.args.get("pre_id")

    if not cur_id:
        return "图片验证码号不全"

    # 2.生成图片验证码
    name, text, image_data = captcha.generate_captcha()
    try:
    # 3.保存图片验证码到redis,key,value,max_age
        redis_store.set("image_code:%s"%cur_id,text,constants.IMAGE_CODE_REDIS_EXPIRES)
    # 4.判断是否有上一个图片验证编码，如果有则删除上次
        if pre_id:
            redis_store.delete("image_code:%s"%pre_id)
    except Exception as e:
        current_app.logger.error(e)
        return "图片验证码保存失败"
    # 5.返回图片验证码
    response=make_response(image_data)
    response.headers["Content-Type"]="image/png"

    return response